"Francisco García Salinas"

# Programa de Ingeniería en Robótica y Mecatrónica Unidad Académica de Ingeniería Eléctrica





## Materia:

Microcontroladores

## **Alumnos:**

Guadalupe Silva Rodríguez

Juan Francisco Márquez Lamas

Genaro Reyes López

Admilcar Elías Guerrero González

# Proyecto:

Microcontrolador

Docente:

Remberto Sandoval Aréchiga

Fecha: 23/11/20





Unidad Académica de Ingeniería Eléctrica

#### Resumen

En este proyecto se realizó un microprocesador con arquitectura Harvard con ayuda de la metodología aprendida en el curso la cual consistió en realizar una serie de pasos determinados para llevar un orden y realizar de manera satisfactoria un microprocesador. Primero se realizó el diagrama de caja negra con el cual se dio una idea de cómo funcionaría dicho proyecto, posteriormente se creó un diagrama general con todos los bloques que constituyen al microprocesador (diagrama de caja blanca), de ahí se fue centrando en cada bloque por separado y se creó una serie de decodificaciones para las instrucciones y señales, una vez clara la función de todas nuestras señales y como se comportaban, se procedió al pseudocódigo de cada bloque, de esta manera es más sencillo terminar el código final el cual consistió solo en unir todas las partes en el software de vivado donde se realizaron las pruebas en base al lenguaje de descripción de hardware verilog.





# Unidad Académica de Ingeniería Eléctrica

# Índice

| Resumen                     | 2  |
|-----------------------------|----|
| Índice                      | 3  |
| Introducción                | 5  |
| Requerimientos              | 7  |
| Arquitectura                | 8  |
| Caja negra                  | 8  |
| Caja blanca                 | 9  |
| Control                     | 10 |
| Registros.                  | 13 |
| ALU                         | 16 |
| Direccionamiento            | 18 |
| Saltos                      | 20 |
| PC                          | 23 |
| Implementación y simulación | 24 |
| Simulaciones                | 24 |
| Diagrama síntesis           | 29 |
| Project summary             | 30 |
| Análisis de resultados      | 32 |
| Conclusiones                | 34 |
| Referencias                 | 35 |
| Apéndice A (Códigos)        | 36 |
| Microprocesador.v           | 36 |
| ALU.v                       | 38 |
| Control.v                   | 40 |
| PC.v                        | 42 |
| Registros.v                 | 44 |
| Saltos.v                    | 46 |





| Saltos.v                | 48 |
|-------------------------|----|
| Direccionamiento.v      | 50 |
| Apéndice B (Test Bench) | 52 |
| Microprocesador_TB.v    | 52 |
| ALU_TB.v                | 54 |
| Control_TB.v            | 56 |
| PC_TB.v                 | 57 |
| Registros_TB.v          | 58 |
| Saltos_TB.v             | 60 |
| Direccionamiento TB.v   | 62 |



Unidad Académica de Ingeniería Eléctrica



#### Introducción



Alguna vez te has cuestionado ¿Qué es un microprocesador? En general, el término "microprocesador", remite al dispositivo principal dentro de las computadoras digitales, es decir, al elemento encargado de realizar los cálculos que permiten desde escribir una carta hasta editar una fotografía; desde administrar una nómina hasta platicar en tiempo real con alguna persona

al otro lado del mundo; desde disfrutar una película hasta controlar complejos procesos industriales; en fin, el concepto de "microprocesador" evoca un dispositivo de enorme poder de cálculo, relativamente costoso, que consume mucha potencia y que, por tanto, sólo está al alcance de pocas personas. Sin embargo, la realidad es muy distinta. En un hogar típico, existen una enorme cantidad de microprocesadores, realizando diversas tareas que buscan facilitar la vida diaria de los usuarios. Se encuentran en el televisor, en el equipo de sonido, en el reproductor de DVD, en los teléfonos celulares, y es que esta tecnología se ha abaratado a tal grado que muchas aplicaciones que antes requerían el uso de varios dispositivos individuales,

#### Sabías que...

El primer microprocesador fue el 4004 de Intel, y apareció en el mercado en 1971. Este hecho fue accidental y se debió a un contrato entre la empresa Intel y una compañía japonesa de calculadores para el desarrollo de un circuito integrado para dicho producto. desarrollo no fue aceptado y la firma Intel se decidió por su comercialización. El 4004 era un microprocesador de 4 bits realizado con tecnología PMOS.

ahora se pueden realizar con más facilidad y de manera más económica con la aplicación de un microprocesador, o de su variante, un microcontrolador.

Esto significa que los microprocesadores y microcontroladores se han convertido en parte de la vida diaria, y esto a su vez implica que cualquier persona interesada en el área de la electrónica o del control debe saber cómo funcionan y cómo se aplican estos dispositivos, es por esto que en este proyecto se dio a la tarea de conocer,





Unidad Académica de Ingeniería Eléctrica

#### ARQUITECTURA HARVARD



Figura 1 Arquitectura Harvard

investigar y crear un microcontrolador en base a una arquitectura Harvard, este modelo, es utilizado por los Microcontroladores PIC, tiene la unidad central de proceso (CPU) conectada a dos memorias (una con las instrucciones y otra con los datos) por medio de dos buses diferentes. Una de las memorias contiene solamente las instrucciones del programa (Memoria de Programa), y la otra sólo almacena datos (Memoria de Datos).

Con esto en mente es que se realizó el microprocesador el cual se presenta a continuación. Se espera un buen recibimiento.





Unidad Académica de Ingeniería Eléctrica

## Requerimientos

Se requiere un sistema que sea capaz de realizar los procesos necesarios para construir un microcontrolador de 8 bits guiado por un set de instrucciones y una metodología preestablecida. Dicho sistema debe permitir que los datos que entren puedan ser procesados y manejados a través de éste. El sistema debe contar con dos entradas que indiquen las instrucciones (9 bits) y la entrada de datos (8 bits), asimismo cuenta con cuatro salidas, las cuales definen la dirección de instrucciones (8 bits), la dirección de datos (8 bits), el bus de datos (8 bits) y el bus RW (define lectura o escritura de 1 bit). Adicionalmente se requieren periféricos para hacerla funcionar de forma correcta.





Unidad Académica de Ingeniería Eléctrica

## **Arquitectura**

**Caja negra.** En la *Figura 2* se muestra la caja negra del microprocesador creado. Así como en las *Tabla 1 y 2* la descripción de las entradas y salidas.



Figura 2 Caja negra Microprocesador

Tabla 1 Entradas Microprocesador

| ENTRADAS      | TAMAÑO<br>(BITS) | DESCRIPCION                                                                                                                                             |
|---------------|------------------|---------------------------------------------------------------------------------------------------------------------------------------------------------|
| Datos_Entrada | 8                | Lleva los datos que entran al microprocesador sobre los que se realizan las instrucciones, los cuales son guardados de forma temporal en los registros. |
| Instrucciones | 9                | Lleva las instrucciones que se van a realizar dentro del procesador, según el set de instrucciones.                                                     |
| Clk           | 1                | Señal referente al tiempo, es una señal periódica con frecuencia de 100MHz                                                                              |
| Rst           | 1                | Señal que reinicia el microprocesador configurándolo a un estado inicial.                                                                               |

Tabla 2 Salidas Microprocesador

| SALIDAS                 | TAMAÑO<br>(BITS) | DESCRIPCION                                                                                   |
|-------------------------|------------------|-----------------------------------------------------------------------------------------------|
| Direccion_Instrucciones | 8                | Controla las direcciones de la memoria de instrucciones.                                      |
| Direccion_Datos         | 8                | Controla las direcciones de memoria , de la memoria de datos                                  |
| Salida_Datos            | 8                | Lleva los datos que salen del microprocesador y que serán almacenados en la memoria de datos. |
| R/W                     | 1                | indicador de acción en la memoria, referente a lectura o escritura                            |





Unidad Académica de Ingeniería Eléctrica

**Caja blanca.** En la *Figura 3* se muestra la caja blanca del microprocesador creado. Así como en la *Tabla 3* la descripción de las entradas y salidas.



Figura 3 Caja blanca

Tabla 3 Caja blanca

| BLOQUE           | DESCRIPCIÓN                                                                                                                                                                                                                                                                                                    |
|------------------|----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| Control          | Sincroniza y procesa las instrucciones. Coordina todos los componentes del microprocesador, lo que hace que cada instrucción siga una secuencia apropiada en el momento correcto. Decodifica y/o entiende las instrucciones que provienen de la memoria de instrucciones, y dirige la acción para realizarlas. |
| Registros        | Contiene registros de propósito general, en los que se guardan datos temporales sobre los que se realizan las instrucciones (es decir, los datos u operandos para realizar las operaciones del set de instrucciones).                                                                                          |
| ALU              | Realiza las operaciones aritméticas y lógicas. Del set de instrucciones se refiere a la instrucción MATH y las operaciones que es capaz de realizar son en este caso: suma y resta sobre enteros, corrimientos a derecha e izquierda, negación, AND, OR Y XOR.                                                 |
| Direccionamiento | Controla la memoria de datos, coordinando el bus de dirección a memoria (Direccion_Datos), el bus de salida de datos (Salida_Datos) y el indicador de acción lecutra o escritura (R/W).                                                                                                                        |
| Saltos           | Se encarga de realizar los saltos del PC trátese de saltos condicionales o no condicionales.                                                                                                                                                                                                                   |
| PC               | Lleva la dirección de memoria donde se guarda la la siguiente instrucción a realizar.                                                                                                                                                                                                                          |







**Control.** En la *Figura 4* se muestra el bloque control del microprocesador creado. Así como en las *Tabla 4 y 5* la descripción de las entradas y salidas.



Figura 4 Bloque control

Tabla 4 Entradas Control

| ENTRADAS                 | TAMAÑO<br>(BITS) | DESCRIPCION                                                                                                                                                                                                                                                                                                                                                                                                                  |
|--------------------------|------------------|------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| i_Instrucciones<br>[8:6] | 3                | Para la segmentación de los bits, los bits más significativos corresponden a la instrucción que se realizará.                                                                                                                                                                                                                                                                                                                |
| i_Instrucciones<br>[5:0] | 6                | Los bits restantes se agrupan igualmente en segmentos de 3 bits y según la instrucción de la que se trate el cada grupo corresponde a los argumentos de la instrucción. Por ejemplo, para la primera instrucción los bits menos significativos corresponden a la variable #NUM, y los más significativos (que en todo el código de instrucción serían los intermedios [5:3]) corresponden al registro donde se carga el dato |
| i_Clk                    | 1                | Señal de reloj que sincroniza todos los bloques del sistema, para controlar la duración de las distintas instrucciones.                                                                                                                                                                                                                                                                                                      |
| i_Rst                    | 1                | Señal que restablece el sistema en una configuración inicial                                                                                                                                                                                                                                                                                                                                                                 |

Tabla 5 Salidas Control

| SALIDAS       | TAMAÑO<br>(BITS) | DESCRIPCION                                                                                                                                                                    |
|---------------|------------------|--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| o_Control_PC  | 1                | Actualiza los datos del PC incrementando a 1 para así obtener la siguiente instrucción a realizar.                                                                             |
| o_Control_ALU | 4                | La señal lleva el código de la operación a realizar correspondiente al argumento OP de la instrucción MATH en los bits menos significativos ([2:0]) y el bit más significativo |





### Unidad Académica de Ingeniería Eléctrica

|                            |   | corresponde a la instrucción, es decir,<br>cuando este bit valga 1 entonces se<br>realizará la operación, de lo contrario la<br>ALU no realizará ninguna operación                             |
|----------------------------|---|------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| o_Control_Direccionamiento | 5 | Salida dirigida a direccionamiento la cual lleva el código de la instrucción que se va a ejecutar y el código del dato que se va a almacenar cuando se trata de un direccionamiento inmediato. |
| O_Control_Saltos           | 4 | Señal que controla la ejecución de saltos, lleva el código de la condición que se está revisando [2:0] y el código de instrucción para verificar si va a realizar la acción o no.              |
| O_Control_Registros        | 1 | Señal dirigida a registros, la cual controla la carga de los datos provenientes de PC y ALU.                                                                                                   |

Descripción funcional

```
Pseudocódigo:
1/0
[8:0] i_Instrucciones;
i_Rst;
i_Clk;
o_Control_PC;
[2:0]o_Control_ALU;
[4:0]o_Control_Direccionamiento;
[3:0]o_Control_Saltos;
o_Control_Registros;
segmentar i_Instrucciones:
c_in <-- i_Instrucciones ([8:6]);
c_reg <-- i_Instrucciones ([5:3]);
c_arg <-- i_Instrucciones ([2:0]);
INICIO
si hay un flanco positivo de reloj:
if (Rst = 1) then
o Control PC <= 0:
o_Control_ALU <=0;
o_Control_Registros <= 0;
o_Control_Direccionamiento <=0;
o_Control_Saltos <=0;</pre>
else
case (c_in)
001: o_Control_Direccionamiento <= {00,c_arg}; o_Control_ALU <= {0,c_arg}; o_Control_Saltos <
c_arg};
                   o_Control_PC <=1; o_Control_Registros <= x;
010: o_Control_Direccionamiento \leftarrow {01,c_arg}; o_Control_ALU \leftarrow {0,c_arg}; o_Control_Saltos \leftarrow {0,
c_arg};
                   o_Control_PC <=1; o_Control_Registros <= x;
```





```
011: o_Control_Direccionamiento <= {10,c_arg}; o_Control_ALU <= {0,c_arg}; o_Control_Saltos <
                 o_Control_PC <=1; o_Control_Registros <= x;
100: o_Control_Direccionamiento <= \{11,c\_arg\}; o_Control_ALU <= \{0,c\_arg\}; o_Control_Saltos <= \{0,c\_arg\};
                 o_Control_PC <=1; o_Control_Registros <= x;
101: o_Control_Direccionamiento <= {00,c_arg}; o_Control_ALU <= {0,c_arg}; o_Control_Saltos <= {0,
                 o_Control_PC <=1; o_Control_Registros <= x;
110: o_Control_Direccionamiento <= {00,c_arg}; o_Control_ALU <= {1,c_arg}; o_Control_Saltos <= {0,
c_arg};
                o_Control_PC <=1; o_Control_Registros <= 0;
111: o_Control_Direccionamiento <= {00,c_arg}; o_Control_ALU <= {0,c_arg}; o_Control_Saltos <= {1,
c_arg};
                o_Control_PC <=0; o_Control_Registros <= 1;
default: o_Control_Direccionamiento
                                                                                                                           <= o_Control_Direccionamiento;</p>
                                                                                                                                                                                                                                               o_Control_ALU
o_Control_ALU;
                          o_Control_Saltos <= o_Control_Saltos; o_Control_PC <=1; o_Control_Refistros <= x;
endcase
FIN
```





Unidad Académica de Ingeniería Eléctrica

**Registros.** En la *Figura 5* se muestra el bloque control del microprocesador creado. Así como en las *Tabla 6* la descripción de las entradas y salidas.



Figura 5 Bloque Registros

Tabla 6 Entradas de Registros

| ENTRADAS            | TAMAÑO<br>(BITS) | DESCRIPCION                                                                                                                                                                                                                                                                                                                                                             |
|---------------------|------------------|-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| i_Datos_Entrada     | 8                | Trae los datos que se cargarán en los registros provenientes de la memoria de datos, los cuales son leídos cuando se ejecuta la instrucción 2                                                                                                                                                                                                                           |
| i_Resultado_ALU     | 8                | Entrada que proviene de la ALU, la cual trae el resultado de la operación que previamente realizó la ALU y que será acumulada en el registro R0.                                                                                                                                                                                                                        |
| i_Intrucciones      | 9                | Entrada proveniente de la memoria de instrucciones, la cual trae el código de la instrucción que se va a ejecutar [8:6], el código del registro que se va a manipular [5:3] y el código del dato que se va a cargar [2:0], en esta caso para la primera instrucción. para cada una de las instrucciones los bits [5:0] corresponden a los argumentos de la instrucción. |
| i_Direccion_PC      | 8                | Entrada proveniente de PC, la cual trae el valor actual de PC la cual será guardada en el registro R7 cuando se realice la condición 1 correspondiente a los saltos.                                                                                                                                                                                                    |
| i_Clk               | 1                | Señal referente al tiempo, es una señal periódica con frecuencia de 100MHz                                                                                                                                                                                                                                                                                              |
| i_Rst               | 1                | Señal que restablece el sistema en una configuración inicial                                                                                                                                                                                                                                                                                                            |
| I_Control_Registros | 1                | Señal proveniente de control con la que se controla la carga de los valores provenientes de PC y ALU respectivamente.                                                                                                                                                                                                                                                   |

Tabla 7 Salidas Registros





Unidad Académica de Ingeniería Eléctrica

| SALIDAS           | TAMAÑO<br>(BITS) | DESCRIPCION                                                                                                                                                                                                                                                                                                               |
|-------------------|------------------|---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| o_Operandos       | 16               | Salida dirigida a la ALU, donde se llevan los datos para que esta realice alguna operación, los cuales se segmentan de la siguiente forma: los bits menos significativos [7:0] corresponden al dato en registro R0 y los bits más significativos [15:8] corresponden al dato guardado en el registro RX                   |
| o_DireccionDato   | 16               | Salida dirigida a direccionamiento, la cual lleva el dato que se va a almacenar en la memoria y la dirección donde se va a realizar la carga, segmentados de tal forma que los bits más significativos corresponden a la dirección de memoria y los bits menos significativos corresponden al dato que se va a almacenar. |
| o_Direccion_Salto | 8                | Salida de dirigida a PC la cual lleva la dirección de memoria a la que va a saltar PC si se cumple una condición.                                                                                                                                                                                                         |

## Descripción funcional

### Pseudocódigo: 1/0 i\_Rst; i\_Clk; i\_Direccion\_PC; i\_Resultado\_ALU; i\_Control\_Registros; i\_Datos\_Entrada; i\_Instrucciones; o\_Direccion\_Salto; o\_Operandos; o\_DireccionDato; señales: c\_ins <= i\_Instrucciones[8:6];</pre> c\_RX <= i\_Instrucciones[5:3];</pre> c\_RY <= i\_Instrucciones[2:0];</pre> reg[7:0] BancoRegistros [0:7]; interger i; INICIO si hay un flanco positivo de reloj: $si (i_Rst = 1) entonces$ inicia ciclo for: for(i = 0; i < 8; i = i+1)BancoRegistros[i] <= 0; termina el ciclo for de lo contrario:





```
case(i_Ins)
001: BancoRegistros [c_RX] <= {00000, c_RY};
010: o_DireccionDato <= { BancoRegistros[c_RY], BancoRegistros[c_RX]};
     if (i_Clk == 1) entonces: BancoRegistros[c_RX] <= i_Datos_Entrada;
011: o\_DireccionDato <= \{BancoRegistros[c\_RX], BancoRegistros[c\_RY]\};
100: o_DireccionDato <= { BancoRegistros[c_RX], BancoRegistros [c_RY];
101: BancoRegistros [c_RX] <= BancoRegistros[c_RY];
110: o_Operandos <= { BancoRegistros[c_RX], Banco_Registros[000] };
111: o_Salto_PC <= BancoRegistros[c_RX];
default: // sin operacion o_DireccionDatos <= o_DireccionDato; o_Salto_PC <= o_Salto_PC;
o_Operandos <= o_Operandos;</pre>
termina case
si(i_control_Registros =0) entonces
BancoRegistros[111] <= i_Direccion_PC;
de lo contrario:
BancoRegistros[000] <= i_Resultado_ALU;
termina bloque de reloj
FIN
```





Unidad Académica de Ingeniería Eléctrica

**ALU.** En la Figura 6 se muestra el bloque ALU. En las tablas se muestra la descripción de las entradas y salidas.



Figura 6 Bloque ALU

| ENTRADAS     | TAMAÑO<br>(BITS) | DESCRIPCION                                                                                                                                                                                                                                                                         |
|--------------|------------------|-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| i_Control_Op | 4                | Entrada proveniente de control, la cual contiene el código de la operación a realizar en los bits menso significativos [2:0], y en el bit más significativo corresponde a la instrucción [3] si este vale 1 entonces la ALU realiza la instrucción de lo contrario no ejecuta nada. |
| i_Opreandos  | 16               | Entrada procedente de registros, la cual trae los datos de los operandos necesarios para realizar la operación, donde los bits menos significativos [7:0] corresponden al registro R0 y los bits más significativos [15:0] corresponden al registro RX.                             |

| SALIDAS           | TAMAÑO<br>(BITS) | DESCRIPCION                                                                                                                         |
|-------------------|------------------|-------------------------------------------------------------------------------------------------------------------------------------|
| o_Resultado       | 8                | Salida dirigida a registros para guardar el resultado que<br>se obtuvo de la operación realizada, la cual será<br>almacenada en R0. |
| o_Banderas_Estado | 3                | Salida dirigida a saltos para informar si se levantó alguna bandera de condición.                                                   |





### Unidad Académica de Ingeniería Eléctrica

#### Descripción funcional

```
Pseudocódigo:
1/0
[3:0] i_Control_ALU;
[15:0] i_Operandos;
[7:0] o_Resultado;
[2:0]o_Banderas_Estado;
señales internas
reg Z;
reg N;
reg C;
reg signed [8:0]Resultado;
INICIO
SEGMENTAR:
Operacion <-- i_Control_ALU([2:0]);
Control<-- i_Control_ALU([3]);
reg signed op2 <-- i_Operandos([15:8]);</pre>
reg signed op1 <-- i_Operandos([7:0]);</pre>
if (Control)
case (Operacion)
000 : Resultado <-- op1+op2;
001 : Resultado <-- op1-op2;
010 : Resultado <-- op1<<op2;
011 : Resultado <-- op1>>op2;
100 : Resultado <-- not(op2);
101 : Resultado <-- op1 & op2;
110 : Resultado <-- op1 | op2;
default : Resultado <-- op1 ^ op2;
endcase
assign Z = (Resultado = 0)? 1'b1: 1'b0;
assign N = (Resultado[8] == 1) ? 1 b1 : 1 b0;
assign C = (Resultado[8] == resultado[7]) ? 1'b0: 1'b1;
o_Banderas_Estado <= { Z, N, C };
o_Resultado <= Resultado[7:0];</pre>
FIN
```





Unidad Académica de Ingeniería Eléctrica

**Direccionamiento.** En la *Figura 7* se muestra el bloque control. Así como en las *Tabla 8 y 9* la descripción de las entradas y salidas.



Figura 7 Bloque direccionamiento

Tabla 8 Entradas Direccionamiento

| ENTRADAS        | TAMAÑO<br>(BITS) | DESCRIPCION                                                                                                                                                                                                                                                                                                                |
|-----------------|------------------|----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| i_DireccionDato | 16               | Entrada proveniente de registros la cual trae el código del dato que se va almacenar en la memoria, así como la dirección donde se realizará el almacenamiento, donde los bits más significativos corresponden a la dirección de memoria y los menos significativos corresponden al código del dato que se va a almacenar. |
| i_Control_Direc | 5                | Entrada procedente de control, la cual trae el código de la instrucción que se desea ejecutar y el valor del dato que se va almacenar en la memoria cuando se trata de un modo de direccionamiento inmediato.                                                                                                              |

Tabla 9 Salidas Direccionamiento

| SALIDAS           | TAMAÑO<br>(BITS) | DESCRIPCION                                                                                                   |
|-------------------|------------------|---------------------------------------------------------------------------------------------------------------|
| o_Direccion_Datos | 8                | Salida dirigida a la memoria de datos donde se lleva la dirección de memoria donde se va a almacenar el dato. |
| o_Salida_Datos    | 8                | Salida dirigida a la memoria de datos donde se lleva el dato que se va a almacenar.                           |
| o_R/W             | 1                | Salida dirigida a la memoria para indicar si la acción que se realiza en esta es de lectura o escritura       |





#### Unidad Académica de Ingeniería Eléctrica

#### Descripción funcional

```
Pseudocódigo:
1/0
[4:0] i_Control_Direc;
[15:0] i_DireccionDato;
[7:0] o_Direccion_Datos;
[7:0] o_Salida_Datos;
R/W;
INICIO
//segmentar:
señales internas
[1:0]c_in \le i_Control_Direc[4:3];
[2:0]c\_dat \le i\_Control\_Direc[2:0];
00: // no hay operación; o_Direccion_Datos <= z; o_Salida_Datos <= z; R/W <= z;
01: o_Direccion_Datos <= i_Registros[15:8]; R/W <= 1; // lectura
10: o_Direccion_Datos <= i_Registros[15:8]; o_Salida_Datos <= {00000, c_dat}; R/W <= 0;
11: o_Direccion_Datos <= i_Registros[15:8]; o_Salida_Datos <= i_Registros[7:0]; R/W <= 0;
endcase
FIN
```





Unidad Académica de Ingeniería Eléctrica

**Saltos.** En la *Figura 5* se muestra el bloque control del microprocesador creado. Así como en las *Tabla 6* la descripción de las entradas y salidas.



Figura 8 Bloque Saltos

Tabla 10 Entradas Saltos

| ENTRADAS        | TAMAÑO<br>(BITS) | DESCRIPCION                                                                                                                                                       |
|-----------------|------------------|-------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| i_BanEstado     | 3                | Entrada proveniente de la ALU la cual indica si se ha levantado alguna bandera.                                                                                   |
| i_Control_Salto | 4                | Entrada procedente de control, la cual trae el código de la condición que se desea verificar [2:0] y la indicación para que el bloque realice la instrucción [3]. |

Tabla 11 Salidas Saltos

| SALIDAS    | TAMAÑO<br>(BITS) | DESCRIPCION                                                                                                                                                                                                                                                                                |
|------------|------------------|--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| o_Salto_Pc | 2                | Salida dirigida a PC para indicar que se va a realizar un salto. Donde las indicaciones se presentan de la siguiente forma: cuando: se cumple una condición o_PC vale 01. se cumple la condición 1 (sin condicion a R7), o_PC vale 10 no se cumple la condición que se evalúa o_PC vale 00 |





## Unidad Académica de Ingeniería Eléctrica

#### Descripción funcional

```
Pseudocódigo:
1/0
[3:0] i_Control_Salto;
 [2:0] i_BanEstado;
 [1:0] o_Salto_PC;
INICIO
si (i_Control_Salto[3] = 1) entonces
  Case (i_Control_Salto[2:0])
     000: o_PC <= 01;
     001: o_PC <= 10;
     010:
       si (i_BanEstado(2) = 1) entonces
          o_Salto_PC <= 01;
       de lo contrario
          o__Salto_PC <= 00;
       si(i\_BanEstado(2) = 0) entonces
          o_Salto_PC <= 01;
       de lo contrario
          o_Salto_PC <= 00;
     100:
       si (i\_BanEstado(1) = 1) entonces
         o_Salto_PC <= 01;
       de lo contrario
         o_Salto_PC <= 00;
 101:
```





```
si(i\_BanEstado(1) = 0) entonces
        o_Salto_PC <= 01;
de lo contrario
        o_Salto_PC <= 00;
     110:
      si(i\_BanEstado(0) = 1) entonces
        o_Salto_PC <= 01;
      de lo contrario
        o_Salto_PC <= 00;
    111:
      si (i_BanEstado(0) = 0) entonces
        o__Salto_PC <= 01;
      de lo contrario
        o_Salto_PC <= 00;
termina case
si (i_Control_Salto[3] == 0 ) entonces
     o_Salto_PC <= 11;
FIN
```





Unidad Académica de Ingeniería Eléctrica

**PC.** En la *Figura 5* se muestra el bloque control del microprocesador creado. Así como en las *Tabla 6* la descripción de las entradas y salidas.



Figura 9 Bloque PC

Tabla 12 Entradas PC

| ENTRADAS          | TAMAÑO<br>(BITS) | DESCRIPCION                                                                                                                    |
|-------------------|------------------|--------------------------------------------------------------------------------------------------------------------------------|
| i_Control_Pc      | 1                | Entrada proveniente de control la cual indica que el PC debe actualizar su valor con un incremento de 1.                       |
| i_Control_Saltos  | 2                | Entrada proveniente de saltos la cual indica que<br>se debe realizar un salto en el PC porque se ha<br>cumplido una condición. |
| i_Direccion_Salto | 8                | Entrada proveniente de registros donde se específica la dirección de memoria a la que va a saltar el PC.                       |
| i_Clk             | 1                | Señal referente al tiempo, es una señal periódica con frecuencia de 100MHz                                                     |
| i_Rst             | 1                | Señal que restablece el sistema en una configuración inicial                                                                   |

Tabla 13 Salidas PC

| SALIDAS                   | TAMAÑO<br>(BITS) | DESCRIPCION                                                                                                                                                |
|---------------------------|------------------|------------------------------------------------------------------------------------------------------------------------------------------------------------|
| o_Direccion_Instrucciones | 8                | Salida dirigida a la memoria de instrucciones la cual controla las direcciones de memoria donde se almacena la siguiente instrucción a realizar.           |
| o_Direccion_PC            | 8                | Salida dirigida a los registros la cual lleva el valor actual del PC que se va a almacenar en R7 cuando se cumpla la condición 1 de la instrucción saltos. |







## Implementación y simulación

#### **Simulaciones**

Enseguida presentamos en las ilustraciones 10 y 11 las simulaciones del microprocesador, además en las ilustraciones de la 12 a la 17 son las simulaciones de cada bloque de la estructura del microprocesador diseñado y por último en las ilustraciones 18 y 19 encontraremos las simulaciones de la memoria RAM y ROM.



Figura 10 Simulación microprocesador



Figura 11 Simulación microprocesador







Figura 12 Simulación ALU



Figura 13 Simulación Control







Figura 14 Simulación Direccionamiento



Figura 15 Simulación PC







Figura 16 Simulación Registros



Figura 17 Simulación Saltos







Figura 18 Simulación memoria RAM



Figura 19 Simulación memoria ROM

## Diagrama síntesis

A continuación, en la ilustración 9 se muestra el esquemático generado por vivado del microprocesador sintetizado.



Figura 20





Unidad Académica de Ingeniería Eléctrica

#### **Project summary**

La sección settings (Figura21) especifica las configuraciones básicas que tendrá el diseño de hardware, como se muestra en la *ilustración* X se encuentra el nombre del proyecto, la ruta en la que se encuentra almacenada, la familia del producto sobre la cual se implementa el diseño, el modelo del FPGA (en este caso la basys 3), el nombre del módulo principal, el lenguaje sobre el cual está basado y el lenguaje en el que se simula.



Figura 21 Settings

La sección síntesis (ilustración 22) se encuentra la información referente al proceso de sintetizado, dando a conocer algún error o advertencia durante el proceso.



Figura 22 Synthesis

En la sección *implementación (ilustración 23)* se encuentra lo referente a esto. Se encuentra el estado y los errores y advertencias.



Figura 23 Implementation





Unidad Académica de Ingeniería Eléctrica

La sección *DRC violations* (*ilustración 24*), se refiere a las reglas de diseño, es decir, indica si hay alguna violación en las reglas o algo fuera del estándar que no genera un error como tal.



Figura 24 DRC violations

La sección *utilización* (*ilustración* 25) se refiere a la cantidad de recursos utilizados por parte de la tarjeta.



Figura 25 Utilization

La sección *power (ilustración 26)* se refiere a la información útil sobre el consumo energético de la tarjeta, en otras palabras, qué tanta potencia consumirá y qué tanto incrementará su temperatura al momento de implementar el diseño sobre el FPGA.



Figura 26 Power





Unidad Académica de Ingeniería Eléctrica

#### Análisis de resultados

Se realizó una descripción de un microprocesador de 8 bits con el propósito de conocer tipos de microprocesadores, su estructura básica y especificaciones, además con el uso de un lenguaje de descripción de hardware logramos concluir este proyecto, ya que éste nos ayuda a describir y simular nuestro procesador.

Para realizar este proyecto es necesario seguir una metodología, por lo que para iniciar debemos tener el diseño de caja negra y comprenderlo, es decir conocer la función que realiza cada una de sus entradas y salidas, además de conocer claramente que operaciones puede realizar este procesador así como sus limitantes, comprender esto requiere un conocimiento previo de teoría sobre los procesadores.

Una vez comprendido el diagrama de caja negra, así como sus posibles operaciones y sus limitantes debemos comenzar a realizar el diagrama de caja blanca. Como aprendimos en la teoría cada microprocesador consta de partes básicas que le permiten realizar las operaciones para el cual fue diseñado, como es el caso de la ALU (Unidad Aritmética Lógica), que nos ayuda a realizar las operaciones entre los registros y entre operadores para posteriormente almacenarse dicha operación como un registro.

También otro bloque esencial entre los microprocesadores es el de los registros, este bloque es muy importante ya que en él se almacenarán operaciones que se deseen guardar, así como registros con los cuales realizar operaciones, así como almacenar un conjunto de operaciones cuando se vaya a realizar un salto.

El bloque de control recibe la instrucción y es el encargado de distribuirla, es decir en la instrucción contiene 3 bits dedicados a la operación a realizar, otros 3 bits para el primer operador y los últimos 3 bits para el segundo operador, y es el encargado de controlar lo que la ALU en caso de que la operación a realizar sea dominio de este bloque, si la operación no es del dominio de la ALU controla los bloques a los cuales la instrucción pertenece. Este bloque se encarga del manejo del microprocesador siendo una parte muy importante.

En el bloque de saltos es donde se interrumpen las operaciones actuales para darle paso a una con mayor importancia por ende las instrucciones interrumpidas son guardadas en registro para no perder la continuidad después del salto.

PC es el encargado de contener la dirección de la siguiente instrucción a ejecutar





Unidad Académica de Ingeniería Eléctrica

El bloque direccionamiento se encarga de enviar las direcciones de los datos, así como el bus de datos a la memoria de datos.

Una vez que sabemos los bloques esenciales que llevará nuestro microprocesador debemos realizar el diagrama de caja blanca, con sus conexiones, nombrando también la señal interna que une un bloque con otro, así como su dimensión, es decir el tamaño de bus requerido para esa señal.

Una vez hecho el diagrama de caja blanca procedemos a hacer un pseudocódigo para resolver cómo será el funcionamiento de cada bloque, este seudocódigo debe presentar las características básicas de un lenguaje de descripción de hardware, para mejor entendimiento.

Teniendo todos los pasos anteriores cumplidos procedemos a realizar la codificación en vivado, herramienta que nos ayudará a simular nuestro microprocesador. Al pasar de pseudocódigo a código en verilog verificamos que la lógica con la cual nuestro bloque sea la misma con la cual trabaja nuestra herramienta para evitar errores que al final resultarán en pérdida de tiempo.

Y finalmente realizamos pruebas corroborando su funcionamiento tanto en sus operaciones, así como las limitaciones que teóricamente debe de tener.





Unidad Académica de Ingeniería Eléctrica

#### **Conclusiones**

Finalmente, se obtuvo un microprocesador de 8 bits capaz de realizar operaciones matemáticas y lógicas a un conjunto de datos. Aparentemente, las simulaciones funcionan de forma correcta, por lo tanto, se espera que, al conectarle las memorias externas, tenga un buen funcionamiento. Se propuso realizar una multiplicación con los recursos con los que cuenta el diseño realizado, por ello, se hicieron múltiples intentos, teniendo el resultado final el cual se mostrará en la presentación del presente.

Es muy importante tanto para el diseño como para el desarrollo de un proyecto o sistema que busque cumplir un requisito establecido o una funcionalidad específica el realizarlo de forma ordenada y siguiendo las jerarquías de diseño, en nuestro caso de un sistema digital, el seguimiento del flujo de diseño establecido de forma correcta y prestando la justa atención a cada parte ayuda a evitar posibles complicaciones futuras durante los pasos posteriores hasta completar el desarrollo de todo el sistema, no sólo para evitar complicaciones en algún punto o que si llegase a ocurrir poder identificarlo con facilidad y emplear menos tiempo en resolverlo, sino que además mejora y facilita el trabajo en equipo ayudando a evitar complicaciones tediosas a otros grupos de trabajo para conseguir el fin deseado en tiempo y sin mayores problemáticas esto es algo que también nos ha dejado como aprendizaje en la realización de este proyecto.

Lo más complicado del proceso de elaboración del microprocesador, fue sin duda alguna entender los conceptos clave que trae consigo la metodología que se describe en el presente documento, esto, debido a que se trata de un gran número de conceptos nuevos, y en ocasiones confusos, sin embargo, se concluye que se tuvo un gran enriquecimiento de la materia, que seguro será útil en un futuro profesional. Otra dificultad, fue la realización de las memorias, debido a que se tuvo que implementar esta idea desde cero en poco tiempo.

Finalmente, se tuvo un buen resultado, solo queda consultarlo con el docente y verificar que, efectivamente, funciona de manera correcta.





#### Unidad Académica de Ingeniería Eléctrica

## Referencias

- [ J. A. SAINZ, INTRODUCCIÓN A LOSMICROPROCESADORES, E.U.I.T.I. VITORIA-GASTEIZ:
- 1 CATEDRÁTICO E.U., 2000.

]

- [ Anónimo, «Significados,» 15 02 2019. [En línea]. Available:
- 2 https://www.significados.com/microprocesador/#:~:text=La%20Unidad%20Central%20de%20
- Procesos, hasta%20con%20millones%20de%20transistores.. [Último acceso: 22 11 2020].
- [ R. Camacho, «Computo Integrado,» 2012 04 9. [En línea]. Available:
- 3 http://rcmcomputointegrado.blogspot.com/2012/04/arquitectura-von-neumann.html. [Último
- ] acceso: 22 11 2020].





Unidad Académica de Ingeniería Eléctrica

## **Apéndice A (Códigos)**

#### Microprocesador.v

```
module Microprocesador(
//Declara las entradas y salidas del microprocesador
    input Clk,
               //Entrada señal cuadrada con frecuencia de 100Mhz
    input Rst,
                   //Entrada que reinicia valores a 0
    input [7:0] Datos Entrada,
//Entrada de datos que serán procesados quardados temporalmente
    input [8:0] Instrucciones,
//Entrada que contiene la instrucción a realizar
    output [7:0] Direction Instructiones,
//Salida que controla la dirección de memoria de las instrucciones
    output [7:0] Direction Datos,
//Salida que controla las direcciones de la memoria de datos
    output [7:0] Salida Datos,
//Salida de los datos procesados que son almacenados en la memoria
    output RW
//Salida que indica la acción de la memoria, si es lectura o escritura
  wire Control PC;
                        //Declaración de las conexiones que unirán los
//diferentes bloques del microprocesador
  wire Control L;
  wire [3:0] Control ALU;
  wire [4:0] Control D;
  wire [3:0] Control Salto;
  wire [15:0] Operandos;
  wire [7:0] Resultado;
  wire [7:0] Direccion;
  wire [15:0] Direccion Dato;
  wire [2:0] Banderas;
  wire [1:0] Salto PC;
  wire [7:0] Direction PC;
  Control Bolque 1 (
                           //Conexiones del bloque Control, se conectan
//las entradas y salidas con señales internas o salidas y entradas
    .i Instrucciones (Instrucciones),
    .o Control PC (Control PC),
    .o Control Registros (Control L),
    .o_Control_ALU (Control ALU),
    .o Control Direccionamiento (Control D),
    .o Control Saltos (Control Salto)
  Registros Bloque 2 (
                          //Conexiones del bloque Registros, se conectan
//las entradas y salidas con señales internas o salidas y entradas
    .i Rst (Rst),
```





```
.i Clk (Clk),
    .i Direccion PC (Direccion PC),
    .i Resultado ALU (Resultado),
    .i Control Registros (Control L),
    .i Datos Entrada (Datos Entrada),
    .i Instrucciones (Instrucciones),
    .o Direccion Salto (Direccion),
    .o DireccionDato (Direccion Dato),
    .o Operandos (Operandos)
 );
 ALU Bloque_3 (
                        //Conexiones del bloque ALU, se conectan las
//entradas y salidas con señales internas o salidas y entradas
    .i Control ALU (Control ALU),
    .i Operandos (Operandos),
    .o Resultado (Resultado),
    .o Banderas Estado (Banderas)
 Direccionamiento Bloque 4 ( //Conexiones del bloque Direccionamiento,
//se conectan las entradas y salidas con señales internas o salidas y
//entradas
    .i Control Direc (Control D),
    .i DireccionDato (Direccion Dato),
    .o Direccion Datos (Direccion Datos),
    .o Salida Datos (Salida Datos),
    .o_RW (RW)
 Saltos Bloque 5 ( //Conexiones del bloque Saltos, se conectan las
//entradas y salidas con señales internas o salidas y entradas
    .i Control Salto (Control Salto),
    .i BanEstado (Banderas),
    .o Salto PC (Salto PC)
 );
 PC Bloque 6 (
                        //Conexiones del bloque PC, se conectan las
//entradas y salidas con señales internas o salidas y entradas
    .i Clk (Clk),
    .i Rst (Rst),
    .i Control PC (Control PC),
    .i Control Saltos (Salto PC),
    .i Direccion Salto (Direccion),
    .o Direccion Instrucciones (Direccion Instrucciones),
    .o Direccion PC (Direccion PC)
 );
endmodule
                //Finaliza el bloque Microprocesador
```





### Unidad Académica de Ingeniería Eléctrica

#### ALU.v

```
module ALU(
// Se declaran las entradas y salidas del bloque ALU
    input [3:0] i Control ALU,
// Entrada proveniente del bloque control
    input [15:0] i Operandos,
//Entrada que carrea los operandos desde el bloque registros
    output [7:0] o Resultado,
//Salida que arroja el resultado de la operacion aritmética lógica
    output [2:0] o Banderas Estado
// Salida que indica el estado de las banderas
    );
    reg signed [8:0] Resultado = 0;
//Registro que guarda temporalemente el valor del resultado
    wire signed [7:0]op1;
//Cables que conectan las partes del bloque
    wire signed [7:0]op2;
    wire [2:0] Operacion;
    wire Control;
    wire Z;
    wire N;
    wire C;
    assign Control = i Control ALU[3];
// En esta sección se asinan los valores de un elemento a otro
    assign Operacion = i Control ALU[2:0];
    assign op1 = i_Operandos[7:0];
    assign op2 = i Operandos[15:8];
always @(i Control ALU or Operacion or op1 or op2)
//Bloque que indica el proceso donde se hacen las operaciones
begin
    if (Control)
                   //Si existe señal desde control
    begin
        case (Operacion)
            3'b000: Resultado <= op1 + op2; //Suma de operandos
            3'b001: Resultado <= op1 - op2; //Resta de operandos
            3'b010: Resultado <= op1 << op2; //Corrimiento a la izquierda
            3'b011: Resultado <= op1 >> op2; //Corrimiento a la derecha
            3'b100: Resultado <= ~op2;</pre>
                                             //Negación del operando
            3'b101: Resultado <= op1 & op2; //Operación lógica and
            3'b110: Resultado <= op1 | op2; //Operación lógica or
            default: Resultado <= op1 ^ op2; //Operacion lógica xor</pre>
        endcase
    end
    else
        Resultado <= Resultado;
end
assign Z = (Resultado == 0) ? 1'b1 : 1'b0;
//Si el resutlado es 0 la bandera Z toma valor de 1
```





```
assign N = (Resultado[8] == 1) ? 1'b1 : 1'b0;
//Si el bit 8 es 1 el resultado corresponde a un valor negativo
assign C = (Resultado[8] == Resultado[7]) ? 1'b0 : 1'b1;
//Si el resultado del bit 8 es igual al del bit 7 se activa flag carry
assign o_Resultado = Resultado[7:0]; //Resultado de 8 bits
assign o_Banderas_Estado = {Z, N, C};
//Asingna la salida de las banderas
endmodule //Finaliza el bloque
```





### Unidad Académica de Ingeniería Eléctrica

#### Control.v

```
module Control (
    input [8:0] i Instrucciones,
//Entrada en la que se encuentran los argumentos del set de instrucciones
    output o_Control PC,
//Salida al PC para indicar el seguimiento de las instrucciones
    output o Control Registros,
//Salida que manda datos volátiles al bloque registros
    output [3:0] o Control ALU,
//Salida que indica si se se requiere realizar una operación en el bloque
//ALU y la operación
    output [4:0] o_Control_Direccionamiento,
//Salida que lleva el código de la instrucción
    output [3:0] o Control Saltos
//Salida que controla la ejecución de los saltos
    ):
    wire [2:0] c in;
//Declaración de conexiones para llevar a cabo el proceso
    wire [2:0] c reg;
    wire [2:0] c arg;
    wire PC;
    wire Registros;
    wire [3:0] ALU;
    wire [4:0] Direccionamiento;
    wire [3:0] Saltos;
    assign c in = i Instrucciones [8:6];
//Asignación de los valores que toman las conexiones respecto a entradas
    assign c reg = i Instrucciones [5:3];
    assign c arg = i Instrucciones [2:0];
    assign o Control PC = PC;
//Asignación del valor de las salidas con respecto a los datos manejados
en el bloque
    assign o Control Registros = Registros;
    assign o Control ALU = ALU;
    assign o Control Direccionamiento = Direccionamiento;
    assign o Control Saltos = Saltos;
 assign Direccionamiento = (c in == 3'b010) ? {2'b01, c arg} :
//Determina el estado de la salida de direccionamiento
                           (c in == 3'b011) ? {2'b10, c arg} :
                            (c in == 3'b100) ? {2'b11, c arg} : 5'b0;
 assign PC = (c_in == 3'b111) ? 1'b0 : 1'b1;
//Asigna valor a PC dependienod de c in
 assign Registros = (c in == 3'b110) ? 1'b1 : 1'b0;
 assign ALU = (c in == 3'b110) ? {1'b1,c arg} :
```









### Unidad Académica de Ingeniería Eléctrica

#### PC.v

```
module PC(
    input i Clk,
                     //Entrada de reloj de 100MHZ
    input i Rst,
//Entrada de reseteo, los valores toman valores de 0
    input i Control PC,
//Entrada que va del bloque control al PC para direccionar las
instrucciones
    input [1:0] i_Control_Saltos,
//Entrada que indica si la instruccion se salta
    input [7:0] i Direccion Salto,
//Entrada que indic la dirección del salto
    output [7:0] o Direccion Instrucciones,
//Salida al exterior del micro que indica la dirección de las
instrucciones
    );
    wire Control;
//Declaración de cables y registros para llevar a cabo el funcionamiento
    wire [1:0] Saltos;
    wire [7:0] Direccion Salto;
    wire [7:0] Direction PC;
    reg [7:0] contador = 0;
assign Control = i Control PC;
//Asigna los valores a los registros y salidas
assign Saltos = i Control Saltos;
assign Direccion Salto = i Direccion Salto;
//assign o Direccion Instrucciones = Direccion Instrucciones;
assign o_Direccion_PC = Direccion PC;
always @(posedge i Clk)
                       //Determina el proceso que se llevará a cabo
begin
     if(i Rst) begin
//Si la señal de reset es 1, el contador se reinicia
       contador <= 0;
     end
     else begin
         if(Control) begin
           contador <= contador +1;</pre>
//Incrementa el contador cada que se mande señal de control
           if (contador == 255)
//Reinicia el contador cuando se llegue al límite
               contador <= 0;</pre>
         end
         else begin
           case (Saltos)
               2'b00:begin
// Suma uno más a la cuenta porque se cedió el salto
```





```
contador <= contador +1;</pre>
                     if (contador == 255)
                         contador <= 0;</pre>
                 end
                 2'b01:begin
//El contador toma el valor de la direccion del salto
                     contador <= Direccion Salto;</pre>
                 end
                 2'b10:begin
//El contador suma un numero más a la cuenta
                     contador <= contador +1;</pre>
                     if (contador == 255)
                         contador <= 0;</pre>
                 end
            endcase
         end
     end
end
assign o Direccion Instrucciones = contador;
//Asigna el valor del contador a la salida de dirección de instrucciones
assign Direccion_PC = (Saltos == 2'b10) ? o_Direccion_Instrucciones :
             //Selecciona el valor de la dirección de direccion PC
dependiendo del valor del estado de saltos
endmodule
                //Termina el bloque
```





### Unidad Académica de Ingeniería Eléctrica

### Registros.v

```
module Registros(
    input i Rst,
                       //Entrada de reloj de 100MHZ
    input i Clk,
//Entrada de reseteo, los valores toman valores de 0
    input [7:0] i Direccion PC,
//Entrada que proviene de PC que trae el estado del bloque guardada en R7
    input [7:0] i Resultado ALU,
                                  //Entrada que proviene de ALU que
lleva el valor del resultado se guarda en RO
    input i Control Registros,
//Entrada que controla la carga de valores
    input [7:0] i_Datos_Entrada,
//Entrada que contiene los datos que provienen de la memoria
    input [8:0] i Instrucciones,
//Entrada que contiene el códogo de la instrucción a ejecutar
    output [7:0] o Direccion Salto,
//Salida que indica a PC si se va a saltar de instrucción
    se almacena en la memoria para realizar la carga
    output [15:0] o Operandos //Salida que va a la ALU que
contiene los operandos de los que se va a hacer la operación
reg[7:0] BancoRegistros [0:7];
//Se declara un registro para almacenar datos temporalmente
integer i;
                      //Se declaran conexiones
wire [2:0] c Ins;
wire [2:0] c RX;
wire [2:0] c RY;
assign c_Ins = i_Instrucciones[8:6];
//Se asigna a cada conexion a un segmentento del bus de instrucciones
assign c RX = i Instrucciones[5:3];
assign c RY = i Instrucciones[2:0];
always@(posedge i Clk)
//Cada que el reloj tiene un flanco positivo se ejecuta el proceso
begin
      if (i Rst)
//Cuando Rst es 1, el contenido de registros se hace cero
       for(i = 0; i <8; i = i+1)</pre>
           BancoRegistros[i] <= 0;</pre>
      else begin
        case(c Ins)
                       //Dependiendo de la instrucción, el banco de
registros reacciona de distinta forma
          3'b001: BancoRegistros [c_RX] <= {5'b00000, c_RY};</pre>
          3'b010: BancoRegistros[c RX] <= i Datos Entrada;
          3'b101: BancoRegistros [c RX] <= BancoRegistros[c RY];</pre>
          3'b111: begin
```





```
if (c RY == 3'b001)
                        BancoRegistros[7] <= i Direccion PC;</pre>
                  end
        endcase
        if (i Control Registros)
//Guarda los datos del resultado en banco de registros
            BancoRegistros[000] <= i Resultado ALU;</pre>
       end
end
assign o Direccion Salto = (c Ins == 3'b111) ? BancoRegistros[c RX] :
        // La direccion de salto toma el valor del banco de registros o
ceros dependiendo de c Ins
assign o Operandos = (c Ins == 3'b110) ? {BancoRegistros[c RX],
BancoRegistros[000]} : 16'b0;
assign o_DireccionDato = (c_Ins == 3'b010) ? { BancoRegistros[c_RY],
BancoRegistros[c RX] } :
//Asigna valor de la salida del direccionamiento del dato
                          (c Ins == 3'b011) ? { BancoRegistros[c RX],
BancoRegistros[c RY] } :
                          (c Ins == 3'b100) ? { BancoRegistros[c RX],
BancoRegistros[c RY] } : 16'b0;
                //Finaliza el bloque Registros
```





### Unidad Académica de Ingeniería Eléctrica

#### Saltos.v

```
module Saltos(
               //Declara las entradas y salidas del bloque
   condición a verificar
   input [2:0] i BanEstado,
                          //Entrada para indicar la
presencia de banderas
                              //Salida que indica a PC que se
   output [1:0] o Salto PC
va a realizar un salto
   );
   wire [3:0] Control; //Se declaran las conexiones para el
movimiento de datos
   wire [2:0] Estado;
   reg [1:0] PC;
                   //Registro que envia estado al PC
   assign o Salto PC = PC;  //Asigna los valores de salida respecto
al registro
   valor de entradas
   assign Estado = i BanEstado;
always @(i Control Salto or i BanEstado) //Cada que ocurra un
salto desde el bloque control o el estado de banderas cambie
begin
   if(Control[3]) begin
      case (Control[2:0])
                            //Case para definir el estado que
tomará la salida hacia PC
          3'b001:begin
             PC <= 2'b10;
          end
          3'b010:begin
             if(Estado[2]) begin
                 PC <= 2'b01;
             end
             else begin
                PC \le 2'b00;
             end
          end
          3'b011:begin
             if(!Estado[2]) begin
                 PC <= 2'b01;
             end
             else begin
                PC <= 2'b00;
             end
          end
          3'b100:begin
```



end

# Universidad Autónoma de Zacatecas



```
if(Estado[1]) begin
                    PC <= 2'b01;
                else begin
                    PC <= 2'b00;
                end
            end
            3'b101:begin
                if(!Estado[1]) begin
                    PC <= 2'b01;
                else begin
                    PC <= 2'b00;
                end
            end
            3'b110:begin
                if(Estado[0]) begin
                    PC <= 2'b01;
                end
                else begin
                    PC <= 2'b00;
                end
            end
            3'b111:begin
                if(!Estado[1]) begin
                    PC <= 2'b01;
                end
                else begin
                    PC <= 2'b00;
                end
            end
            default: begin
                PC <= 2'b01;
            end
        endcase
    end
    else begin
        PC <= 2'b11;
    end
endmodule
                //Finaliza el bloque
```





## Unidad Académica de Ingeniería Eléctrica

#### Saltos.v

```
//Declara las entradas y salidas del bloque
module Saltos(
    input [3:0] i Control Salto,
//Entrada que carrea el código de condición a verificar
    input [2:0] i BanEstado,
//Entrada para indicar la presencia de banderas
    output [1:0] o Salto PC
//Salida que indica a PC que se va a realizar un salto
    );
    wire [3:0] Control;
//Se declaran las conexiones para el movimiento de datos
    wire [2:0] Estado;
    reg [1:0] PC;
                        //Registro que envia estado al PC
    assign o Salto PC = PC;
//Asigna los valores de salida respecto al registro
    assign Control = i Control Salto;
//Las conexiones toman el valor de entradas
    assign Estado = i BanEstado;
always @(i Control Salto or i BanEstado)
                                                //Cada que ocurra un
salto desde el bloque control o el estado de banderas cambie
begin
    if(Control[3]) begin
        case (Control[2:0])
//Case para definir el estado que tomará la salida hacia PC
            3'b001:begin
                PC <= 2'b10;
            end
            3'b010:begin
                if(Estado[2]) begin
                    PC <= 2'b01;
                else begin
                    PC <= 2'b00;
                end
            end
            3'b011:begin
                if(!Estado[2]) begin
                    PC <= 2'b01;
                end
                else begin
                    PC <= 2'b00;
                end
            end
```



end

# Universidad Autónoma de Zacatecas



```
3'b100:begin
                if(Estado[1]) begin
                    PC <= 2'b01;
                end
                else begin
                    PC <= 2'b00;
                end
            end
            3'b101:begin
                if(!Estado[1]) begin
                    PC <= 2'b01;
                end
                else begin
                    PC \le 2'b00;
                end
            end
            3'b110:begin
                if(Estado[0]) begin
                    PC <= 2'b01;
                end
                else begin
                    PC <= 2'b00;
                end
            end
            3'b111:begin
                if(!Estado[1]) begin
                    PC <= 2'b01;
                end
                else begin
                    PC <= 2'b00;
                end
            end
            default: begin
                PC <= 2'b01;
            end
        endcase
    end
    else begin
        PC <= 2'b11;
    end
endmodule
                //Finaliza el bloque
```





### Unidad Académica de Ingeniería Eléctrica

#### Direccionamiento.v

```
module Direccionamiento (
//Se declaran las entradas y salidas del bloque
    input [4:0] i Control Direc,
//Entrada que trae la instrucción que se ejecutará
    registros donde viene la dirección de donde se quardará el dato
   la que se define la dirección
   output [7:0] o Salida Datos,
//Va a la memoria y lleva el dato a almacenar
   output reg o RW
                         //Define si se trata de lectura o escritura
 );
 wire [1:0] c in;
                      //Declara conexiones entre bloques
 wire [2:0] c dat;
 wire [7:0] Datos;
 reg [7:0] Salida Datos = 0;
reg [7:0] Direction Datos = 0;
 assign c in = i Control Direc[4:3];
//Segmenta la entrada de i Control Direc
 assign c dat = i Control Direc[2:0];
 assign o Salida Datos = Salida Datos;
//Asigna los datos de los registros a las salidas
 assign o Direccion Datos = Direccion Datos;
assign Datos = (c dat == 3'b000) ? 8'b00000001:
                                                     //1
              (c dat == 3'b001) ? 8'b00000010:
                                                     //2
              (c dat == 3'b010) ? 8'b00000100:
                                                     //4
              (c dat == 3'b011) ? 8'b00001000:
                                                     //8
              (c dat == 3'b100) ? 8'b00010000:
                                                     //16
              (c_dat == 3'b101) ? 8'b00100000:
                                                     //32
              (c dat == 3'b110) ? 8'b01000000:
                                                     //64
              (c dat == 3'b111) ? 8'b100000000: 8'h00; //128:00
always@(c in)
begin
    if (c in == 2'b00)
       Direccion Datos <= Direccion Datos;</pre>
//La salida de datos toma el valor de direccionamiento
       Direccion Datos <= i DireccionDato[15:8];</pre>
//La salida de datos toma el valor de direccionamiento
    case(c in)
    2'b01: begin
           Salida Datos <= i DireccionDato[7:0];</pre>
//La salida de datos toma el valor de direccionamiento
```





```
o RW <= 0;
                          //La salida RW indica que es lectura
          end
   2'b10: begin
           Salida Datos <= Datos;
                        //La salida RW indica que es escritura
           o RW <= 1;
          end
   2'b11: begin
           Salida Datos <= i DireccionDato[7:0];</pre>
//La salida de datos toma el valor de direccionamiento
           o_RW <= 1;
//La salida RW indica que es escritura
   default: begin
             Salida Datos <= Salida Datos;
             o RW <= 0; //La salida RW indica que es lectura
            end
   endcase
end
endmodule
```





Unidad Académica de Ingeniería Eléctrica

# **Apéndice B (Test Bench)**

### Microprocesador\_TB.v

```
module Microprocesador TB(
    reg Clk;
    reg Rst;
    reg [7:0] Datos Entrada;
    reg [8:0] Instrucciones;
    wire [7:0] Direccion Instrucciones;
    wire [7:0] Direction Datos;
    wire [7:0] Salida Datos;
    wire RW;
    Microprocesador DUT (
         .Clk (Clk),
         .Rst (Rst),
         .Datos Entrada (Datos Entrada),
         .Instrucciones (Instrucciones),
         .Direccion Instrucciones (Direccion Instrucciones),
         .Direccion Datos (Direccion Datos),
         .Salida Datos (Salida Datos),
         .RW (RW)
    );
  initial
  begin
    Clk <= 0;
    Rst <= 1;
    Datos Entrada <= 8'b00001111;</pre>
    Instrucciones <= 9'b000101011;</pre>
    #20 Rst <= 0;
    Datos Entrada <= 8'b00001110;</pre>
    Instrucciones <= 9'b000101111;</pre>
    #10;
    Datos_Entrada <= 8'b00001110;</pre>
    Instrucciones <= 9'b001111011;</pre>
    Datos Entrada <= 8'b00001110;</pre>
    Instrucciones <= 9'b010001000;</pre>
    Datos Entrada <= 8'b00001110;</pre>
    Instrucciones <= 9'b001100101;</pre>
    #10;
    Datos Entrada <= 8'b00001110;</pre>
    Instrucciones <= 9'b001011110;</pre>
```





Unidad Académica de Ingeniería Eléctrica

```
#10;
  Datos Entrada <= 8'b00001110;</pre>
  Instrucciones <= 9'b110011000;</pre>
  #10;
  Datos Entrada <= 8'b00001110;</pre>
  Instrucciones <= 9'b101010000;</pre>
  Datos Entrada <= 8'b00001110;</pre>
  Instrucciones <= 9'b110111001;</pre>
  #10;
  Datos Entrada <= 8'b00110010;</pre>
  Instrucciones <= 9'b010101001;</pre>
  #10;
  Datos Entrada <= 8'b00110010;</pre>
  Instrucciones <= 9'b011111010;</pre>
  Datos Entrada <= 8'b00110010;</pre>
  Instrucciones <= 9'b110001100;</pre>
  Datos Entrada <= 8'b00110010;</pre>
  Instrucciones <= 9'b111101100;</pre>
  Datos Entrada <= 8'b00110010;</pre>
  Instrucciones <= 9'b100011000;</pre>
  Datos_Entrada <= 8'b00110010;</pre>
  Instrucciones <= 9'b000111111;</pre>
  #10;
  Datos Entrada <= 8'b11111111;</pre>
  Instrucciones <= 9'b010001101;</pre>
  #10;
  Datos Entrada <= 8'b11111111;</pre>
  Instrucciones <= 9'b1111110001;</pre>
  Datos Entrada <= 8'b11111111;</pre>
  Instrucciones <= 9'b000111010;</pre>
  #10;
end
always@(Clk) begin
    #5 Clk <= ~Clk;
end
```

endmodule





### Unidad Académica de Ingeniería Eléctrica

#### ALU\_TB.v

```
module ALU TB(
    );
  reg [3:0] Control;
  reg [15:0] Operandos;
  wire [7:0] Resultado;
  wire [2:0] Banderas;
  ALU DUT (
     .i_Control_ALU(Control),
     .i Operandos (Operandos),
     .o Resultado (Resultado),
     .o_Banderas_Estado(Banderas)
  );
  initial
  begin
    Control <= 0;</pre>
    Operandos <= 0;
    #20;
    Control <= 4'b1000;</pre>
    Operandos[15:8] <= 8'b011111111;
    Operandos[7:0] <= 8'b00000001;
    #20;
    Control <= 4'b1001;</pre>
    Operandos[15:8] <= 8'b11101110;</pre>
    Operandos[7:0] <= 8'b11110111;</pre>
    #20;
    Control <= 4'b1010;</pre>
    Operandos[15:8] <= 8'b00000101;
    Operandos[7:0] <= 8'b00000011;</pre>
    #20;
    Control <= 4'b1011;</pre>
    Operandos[15:8] <= 8'b00000101;
    Operandos[7:0] <= 8'b00000011;</pre>
    #20;
    Control <= 4'b1100;</pre>
    Operandos[15:8] <= 8'b01010101;</pre>
    Operandos[7:0] <= 8'b01010101;</pre>
    #20;
    Control <= 4'b1101;</pre>
    Operandos[15:8] <= 8'b01010101;
    Operandos[7:0] <= 8'b10101010;
    #20;
    Control <= 4'b1110;</pre>
    Operandos[15:8] <= 8'b01010101;</pre>
    Operandos[7:0] <= 8'b10101010;</pre>
    #20;
    Control <= 4'b1111;</pre>
    Operandos[15:8] <= 8'b00000101;
    Operandos[7:0] <= 8'b00000010;
```





```
#20;
Control <= 4'b0010;
Operandos[15:8] <= 8'b00000101;
Operandos[7:0] <= 8'b00000011;
#20;
end
endmodule</pre>
```





Unidad Académica de Ingeniería Eléctrica

#### Control\_TB.v

```
module Control TB(
    reg [8:0] Instrucciones;
    wire PC;
    wire Registros;
    wire [3:0] ALU;
    wire [4:0] Direccionamiento;
    wire [3:0] Saltos;
    Control DUT (
        .i Instrucciones (Instrucciones),
         .o Control PC(PC),
         .o Control Registros (Registros),
         .o Control ALU(ALU),
         .o_Control_Direccionamiento(Direccionamiento),
         .o Control Saltos (Saltos)
    );
    initial begin
         Instrucciones<=9'b010100101;</pre>
         #20;
         Instrucciones<=9'b010100101;</pre>
         Instrucciones<=9'b100010101;</pre>
         #20;
         Instrucciones<=9'b111000110;</pre>
         #20;
         Instrucciones<=9'b110010100;</pre>
         Instrucciones<=9'b0001111111;</pre>
         #20;
    end
```

endmodule





Unidad Académica de Ingeniería Eléctrica

#### PC\_TB.v

```
module PC TB(
    );
    reg Clk;
    reg Rst;
    reg Control;
    reg [1:0] Saltos;
    reg [7:0] Direccion Salto;
    wire [7:0] Direccion Instrucciones;
    wire [7:0] Direction PC;
    PC DUT (
         .i_Clk(Clk),
         .i Rst(Rst),
         .i Control PC(Control),
         .i Control Saltos (Saltos),
         .i Direccion Salto (Direccion Salto),
         .o Direccion Instrucciones (Direccion Instrucciones),
         .o Direccion PC (Direccion PC)
    );
initial
begin
    Clk \le 0;
    Rst \le 1;
    Control <= 1'b1; Saltos <= 2'b00; Direction Salto <= 8'b00000001;
    #20 Rst<=0;
    Control<=1'b1; Saltos<=2'b00; Direction Salto<=8'b00000001;</pre>
    Control<=1'b1; Saltos<=2'b10; Direction Salto<=8'b111111111;</pre>
    Control<=1'b1; Saltos<=2'b01; Direction Salto<=8'b01111000;</pre>
    Control <= 1'b0; Saltos <= 2'b00; Direccion Salto <= 8'b01010101;
    Control <= 1'b0; Saltos <= 2'b01; Direccion Salto <= 8'b111111111;
    Control <= 1'b0; Saltos <= 2'b10; Direccion Salto <= 8'b101010111;
    Control <= 1'b1; Saltos <= 2'b00; Direction Salto <= 8'b01010101;
    #20;
end
always@(Clk)
  #5 Clk <= ~Clk;
endmodule
```





### Unidad Académica de Ingeniería Eléctrica

### Registros\_TB.v

```
module Registros TB (
    reg Rst;
    reg Clk;
    reg [7:0] PC;
    reg [7:0] ALU;
    reg [7:0] Datos_Entrada;
    reg Control;
    reg [8:0] Instrucciones;
    wire [15:0] Direccionamiento;
    wire [7:0] oPC;
    wire [15:0] oALU;
    Registros DUT (
         .i Rst (Rst),
         .i Clk (Clk),
         .i Direccion PC (PC),
         .i Resultado ALU (ALU),
         .i Control Registros (Control),
         .i Datos Entrada (Datos Entrada),
         .i Instrucciones (Instrucciones),
         .o DireccionDato (Direccionamiento),
         .o_Direccion_Salto (oPC),
         .o_Operandos (oALU)
    );
    initial
    begin
        Rst <= 1;
        Clk <= 0;
        PC <= 8'b01010101;
        ALU <= 8'b10101010;
        Control <= 1;
        Datos Entrada <= 8'b11111111;</pre>
        Instrucciones <= 9'b010101011;</pre>
         #20 Rst <= 0;
         Control <= 3'bx;
         Instrucciones <= 9'b001100111;</pre>
         #20;
         Control <= 0;
         Instrucciones <= 9'b010001000;</pre>
        #20;
         Control <= 1;
        Instrucciones <= 9'b1010111100;</pre>
        #20;
        Control <= 0;</pre>
         Instrucciones <= 9'b100100101;</pre>
```





```
#20;
    Control <= 1;
    Instrucciones <= 9'b1101111110;
#20;
    Control <= 1'bx;
    Instrucciones <= 9'b111101000;
    #20;
end

always@(Clk) begin
    #5 Clk <= ~Clk;
end
endmodule</pre>
```





### Unidad Académica de Ingeniería Eléctrica

#### Saltos\_TB.v

```
module Saltos TB(
    reg [3:0] Control;
    reg [2:0] Estado;
    wire [1:0] PC;
    Saltos DUT (
         .i Control Salto (Control),
         .i BanEstado (Estado),
         .o_Salto_PC(PC)
    );
    initial begin
         Control <= 4 'b 00000;
         Estado<=3'b000;
         #20;
         Control <= 4 'b 0001;
         Estado<=3'b001;
         #20;
         Control <= 4 'b 0010;
         Estado<=3'b010;
         #20;
         Control <= 4 'b 0 0 11;
         Estado<=3'b011;
         #20;
         Control<=4'b0100;
         Estado<=3'b100;
         #20;
         Control<=4'b0101;</pre>
         Estado<=3'b101;
         #20;
         Control<=4'b0110;</pre>
         Estado<=3'b110;
         Control<=4'b0111;</pre>
         Estado<=3'b111;
         Control <= 4 'b 1000;
         Estado<=3'b000;
         #20;
         Control <= 4 'b 1001;
         Estado<=3'b001;
         #20;
         Control <= 4 'b 1010;
         Estado<=3'b010;
         #20;
         Control<=4'b1011;</pre>
         Estado<=3'b001;
```





Unidad Académica de Ingeniería Eléctrica

```
#20;
Control<=4'b1100;
Estado<=3'b100;
#20;
Control<=4'b1101;
Estado<=3'b010;
#20;
Control<=4'b1110;
Estado<=3'b001;
#20;
Control<=4'b1111;
Estado<=3'b010;
#20;</pre>
```

endmodule





### Unidad Académica de Ingeniería Eléctrica

### Direccionamiento\_TB.v

```
module Direccionamiento TB(
 reg [4:0] i_Control_Direc;
 reg [15:0] i DireccionDato;
 wire [7:0] o Direccion Datos;
 wire [7:0] o Salida Datos;
 wire o_RW;
 Direccionamiento DUT(
     .i_Control_Direc (i_Control_Direc),
     .i DireccionDato (i DireccionDato),
     .o Direccion Datos (o Direccion Datos),
     .o Salida Datos (o Salida Datos),
      .o RW (o RW)
  );
  initial
  begin
    i Control Direc <= 0;
    i DireccionDato <= 0;
    #20;
    i Control Direc <= 5'b01101;
    i_DireccionDato[15:8] <= 8'b1010101010;</pre>
    i DireccionDato[7:0] <= 8'b01010101;</pre>
    #20;
    i Control Direc <= 5'b10111;</pre>
    i DireccionDato[15:8] <= 8'b00111010;</pre>
    i DireccionDato[7:0] <= 8'b01011100;</pre>
    i Control Direc <= 5'b11011;</pre>
    i DireccionDato[15:8] <= 8'b11110000;
    i DireccionDato[7:0] <= 8'b01111101;</pre>
    #20;
    i Control Direc <= 5'b00101;
    i DireccionDato[15:8] <= 8'b00000011;</pre>
    i DireccionDato[7:0] <= 8'b11111000;</pre>
    #20;
    i Control Direc <= 5'b01000;
    i DireccionDato[15:8] <= 8'b00011100;
    i DireccionDato[7:0] <= 8'b00001111;</pre>
    #20;
    i Control Direc <= 5'b10001;</pre>
    i DireccionDato[15:8] <= 8'b01100111;</pre>
    i DireccionDato[7:0] <= 8'b011010100;</pre>
    #20;
  end
endmodule
```